1
2
3
4
5
6 package spoon.reflect.visitor;
7
8 import spoon.reflect.code.CtExpression;
9 import spoon.reflect.code.CtFieldAccess;
10 import spoon.reflect.code.CtInvocation;
11 import spoon.reflect.code.CtTargetedExpression;
12 import spoon.reflect.code.CtTypeAccess;
13 import spoon.reflect.declaration.CtElement;
14 import spoon.reflect.declaration.CtExecutable;
15 import spoon.reflect.declaration.CtField;
16 import spoon.reflect.declaration.CtMethod;
17 import spoon.reflect.declaration.CtNamedElement;
18 import spoon.reflect.declaration.CtType;
19 import spoon.reflect.declaration.CtTypeMember;
20 import spoon.reflect.path.CtRole;
21 import spoon.reflect.reference.CtExecutableReference;
22 import spoon.reflect.reference.CtFieldReference;
23 import spoon.reflect.reference.CtPackageReference;
24 import spoon.reflect.reference.CtTypeReference;
25 import spoon.support.Experimental;
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52 @Experimental
53 public class ImportConflictDetector extends ImportAnalyzer<LexicalScope> {
54
55 @Override
56 protected LexicalScopeScanner createScanner() {
57 return new LexicalScopeScanner();
58 }
59
60 @Override
61 protected LexicalScope getScannerContextInformation() {
62 return ((LexicalScopeScanner) scanner).getCurrentLexicalScope();
63 }
64
65 @Override
66 protected void handleTargetedExpression(CtTargetedExpression<?, ?> targetedExpression, LexicalScope nameScope) {
67 CtExpression<?> target = targetedExpression.getTarget();
68 if (target == null) {
69 return;
70 }
71 if (targetedExpression instanceof CtFieldAccess<?>) {
72 CtFieldAccess<?> fieldAccess = (CtFieldAccess<?>) targetedExpression;
73 if (target.isImplicit()) {
74
75
76
77
78
79 final CtField<?> field = fieldAccess.getVariable().getFieldDeclaration();
80 if (field != null) {
81 final String fieldName = field.getSimpleName();
82 nameScope.forEachElementByName(fieldName, named -> {
83 if (named instanceof CtMethod) {
84
85 return null;
86 }
87 if (named == field) {
88 return true;
89 }
90
91 target.setImplicit(false);
92 return false;
93 });
94 }
95 }
96 if (!target.isImplicit()) {
97
98 if (target instanceof CtTypeAccess) {
99
100 CtTypeAccess<?> typeAccess = (CtTypeAccess<?>) target;
101 CtTypeReference<?> accessedTypeRef = typeAccess.getAccessedType();
102 checkConflictOfTypeReference(nameScope, accessedTypeRef);
103 }
104 }
105 }
106 }
107
108 @Override
109 protected void handleTypeReference(CtTypeReference<?> ref, LexicalScope nameScope, CtRole role) {
110 if (ref.isImplicit()) {
111
112
113
114
115
116 CtTargetedExpression<?, ?> targetedExpr = getParentIfType(getParentIfType(ref, CtTypeAccess.class), CtTargetedExpression.class);
117 if (targetedExpr instanceof CtInvocation<?>) {
118 CtInvocation<?> invocation = (CtInvocation<?>) targetedExpr;
119 CtExecutableReference<?> importedReference = invocation.getExecutable();
120 CtExecutable<?> importedElement = importedReference.getExecutableDeclaration();
121 if (importedElement == null) {
122
123
124
125 return;
126 }
127 if (importedElement instanceof CtMethod) {
128
129 nameScope.forEachElementByName(importedReference.getSimpleName(), named -> {
130 if (named instanceof CtMethod<?>) {
131
132 if (isSameStaticImport(named, importedElement)) {
133
134 return true;
135 }
136 ref.setImplicit(false);
137 ref.setSimplyQualified(true);
138 return false;
139 }
140
141 return null;
142 });
143 }
144 } else if (targetedExpr instanceof CtFieldAccess<?>) {
145 CtFieldAccess<?> fieldAccess = (CtFieldAccess<?>) targetedExpr;
146 CtFieldReference<?> importedReference = fieldAccess.getVariable();
147 CtElement importedElement = importedReference.getFieldDeclaration();
148 if (importedElement == null) {
149
150
151
152 return;
153 }
154
155 nameScope.forEachElementByName(importedReference.getSimpleName(), named -> {
156 if (named instanceof CtMethod<?>) {
157
158 return null;
159 }
160 if (named == importedElement) {
161
162 return true;
163 }
164
165 ref.setImplicit(false);
166 ref.setSimplyQualified(true);
167 return false;
168 });
169 }
170
171
172 }
173 if (!ref.isImplicit() && ref.isSimplyQualified()) {
174
175
176
177
178 String refQName = ref.getQualifiedName();
179
180 nameScope.forEachElementByName(ref.getSimpleName(), named -> {
181 if (named instanceof CtMethod) {
182
183 return null;
184 }
185 if (named instanceof CtType) {
186 CtType<?> type = (CtType<?>) named;
187 if (refQName.equals(type.getQualifiedName())) {
188
189 return true;
190 }
191 }
192
193 ref.setImplicit(false);
194 ref.setSimplyQualified(false);
195 return false;
196 });
197 }
198 checkConflictOfTypeReference(nameScope, ref);
199 }
200
201
202
203
204
205 private void checkConflictOfTypeReference(LexicalScope nameScope, CtTypeReference<?> typeRef) {
206 if (typeRef == null) {
207 return;
208 }
209 if (!typeRef.isSimplyQualified()) {
210
211
212 if (isPackageNameConflict(nameScope, typeRef)) {
213
214 typeRef.setSimplyQualified(true);
215 if (isSimpleNameConflict(nameScope, typeRef)) {
216
217 typeRef.setImplicit(true);
218 }
219 }
220 } else {
221 if (!typeRef.isImplicit()) {
222 if (isSimpleNameConflict(nameScope, typeRef)) {
223
224
225 typeRef.setSimplyQualified(false);
226 }
227 }
228 }
229 }
230
231 private boolean isPackageNameConflict(LexicalScope nameScope, CtTypeReference<?> typeRef) {
232 String fistPackageName = getFirstPackageQName(typeRef);
233 if (fistPackageName != null) {
234 return Boolean.TRUE == nameScope.forEachElementByName(fistPackageName, named -> {
235 if (named instanceof CtMethod) {
236
237 return null;
238 }
239
240 return Boolean.TRUE;
241 });
242 }
243 return false;
244 }
245
246 private boolean isSimpleNameConflict(LexicalScope nameScope, CtTypeReference<?> typeRef) {
247 String typeQName = typeRef.getQualifiedName();
248
249 return Boolean.TRUE == nameScope.forEachElementByName(typeRef.getSimpleName(), named -> {
250 if (named instanceof CtMethod) {
251
252 return null;
253 }
254 if (named instanceof CtType) {
255 CtType<?> type = (CtType<?>) named;
256 if (typeQName.equals(type.getQualifiedName())) {
257
258 return Boolean.FALSE;
259 }
260
261 }
262
263 return Boolean.TRUE;
264 });
265 }
266
267 private String getFirstPackageQName(CtTypeReference<?> typeRef) {
268 if (typeRef != null) {
269 CtPackageReference packRef = typeRef.getPackage();
270 if (packRef != null) {
271 String qname = packRef.getQualifiedName();
272 if (qname != null && qname.length() > 0) {
273 int idx = qname.indexOf('.');
274 if (idx < 0) {
275 idx = qname.length();
276 }
277 return qname.substring(0, idx);
278 }
279 }
280 }
281 return null;
282 }
283
284
285
286
287
288
289
290
291
292
293 private static boolean isSameStaticImport(CtNamedElement m1, CtNamedElement m2) {
294 if (m1 instanceof CtTypeMember && m2 instanceof CtTypeMember) {
295 if (m1.getSimpleName().equals(m2.getSimpleName())) {
296 CtType<?> declType1 = ((CtTypeMember) m1).getDeclaringType();
297 CtType<?> declType2 = ((CtTypeMember) m2).getDeclaringType();
298
299 return declType1 == declType2;
300 }
301 }
302 return false;
303 }
304 }